# 电子科技大学

嵌入式智能计算研究团队

# 珊瑚-I 启动过程

**ACORAL-I BOOT PROCESS** 



## Revision History

| 版本号 | 内容                    | 日期         | 负责人 |
|-----|-----------------------|------------|-----|
| 0.1 | 开始编写,修改 latex 模板,确定大纲 | 2022.05.15 | 王彬浩 |

## 目 录

| 一章  | Bootloader                     | 1                                                                                                                                                                                                             |
|-----|--------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 1.1 | 什么是 Bootloader                 | 1                                                                                                                                                                                                             |
| 1.2 | loader                         | 2                                                                                                                                                                                                             |
| 二章  | aCoral 启动                      | 5                                                                                                                                                                                                             |
| 2.1 | 启动-第一阶段                        | 5                                                                                                                                                                                                             |
|     | 2.1.1 禁用看门狗                    | 6                                                                                                                                                                                                             |
|     | 2.1.2 关中断                      | 7                                                                                                                                                                                                             |
| 2.2 | 时钟初始化                          | 8                                                                                                                                                                                                             |
|     | 2.2.1 存储空间初始化                  | 10                                                                                                                                                                                                            |
|     | 2.2.2 栈初始化                     | 10                                                                                                                                                                                                            |
|     | 2.2.3 自我拷贝(loader)             | 12                                                                                                                                                                                                            |
|     | 2.2.4 bss 段清零                  | 13                                                                                                                                                                                                            |
|     | 2.2.5 跳转至下一阶段                  | 13                                                                                                                                                                                                            |
| 2.3 | 启动-第二阶段                        | 14                                                                                                                                                                                                            |
| Ξ章  | 链接脚本                           | 15                                                                                                                                                                                                            |
|     | 1.1<br>1.2<br>二章<br>2.1<br>2.2 | -章 Bootloader  1.1 什么是 Bootloader  1.2 loader  二章 aCoral 启动  2.1 启动-第一阶段  2.1.1 禁用看门狗  2.1.2 关中断  2.2 时钟初始化  2.2.1 存储空间初始化  2.2.2 栈初始化  2.2.3 自我拷贝(loader)  2.2.4 bss 段清零  2.2.5 跳转至下一阶段  2.3 启动-第二阶段  章 链接脚本 |

## 第一章 Bootloader

#### 1.1 什么是 Bootloader

我们将 Bootloader 拆开来看,一个是 boot, 一个是 loader,可见 Bootloader 有两个主要特性:

- (1) Boot。Boot 的原意为靴子,在计算机领域引申为启动,也就是说系统启动时会从这里启动,具体一点就是当我们按开机键,cpu 执行的第一条指令就应该是Bootloader 的代码。
  - (2) Loader。Loader 是加载的意思。那自然就可以引出五个问题:
  - i. 加载什么(what)

当然是加载代码程序,这个程序就是我们经常谈到的内核映像——像 linux 内核, window 内核, 亦或是 aCoral 的内核等。

ii. 怎么加载(how)

加载说白了就是复制,将内核代码原封不动的从一个地方复制到另一个地方。

iii. 从哪里加载(from)

自然是从放着内核代码的地方加载了,常见的存储介质有 flash、SD 卡等。

iiii. 加载到哪里去(to)

自然是加载到内核要运行的地方——内存,常见的有 SDRAM、DDR 等。

iiii. 为什么要加载(why)

这就不是一句两句能说清楚的了。简单来讲,因为 flash、SD 卡这些非易失性存储器不能用来运行程序,或者不适合运行程序,但是他们便宜,容量大,断电也不会丢失数据,适合存放程序,我们一般称为外存。而 SDRAM、DDR 这些易失性存储器不能用来在断电的情况下存放程序,但是在通电之后运行速度快,适合运行代码,一般称为内存。所以,上电之后需要通过 loader 来把内核程序从外存复制到内存中,这样内核程序就能正常运行了。

在嵌入式系统中,常见的 Bootloader 有 vivi,uboot ,这些程序都是开机时就启动的,它启动后,会从 flash 或 sd 卡等存储设备中将内核程序代码拷贝到 sdram, 然后执行内核代码。

如果你曾经接触过 vivi、uboot 这些开源的 bootloader,就会发现,这些 bootloader 似乎都没有这么简单,大小也都往往几 MB 往上,这是为什么呢?有两个主要原因:(1)它们都支持多平台,它们都可以看成一个通用的 Bootloader。(2)除了提供上面启动,加载两个功能外,它们还支持更多功能,比如支持各种命令,比如操

作 nandflash,norflash,EEPROM,支持 ftp,tftp,nfs 等网络协议,又或者支持 usb 下载等功能。有些 Bootloader 如 arm 公司的 bootmonitor 还支持文件系统,能以文件系统的方式管理 nandflash,sdcard,compact card 上的数据。有了上面两大类的支持后,Bootloader 不再是纯粹的 Bootloader,都有了操作系统的一些功能,只是不支持操作系统支持的任务切换功能。

aCoral 的 bootloader 其实就是

hal/s3c2440/src/start.S

这个文件。它就没有这么多复杂的功能了,仅仅做了 bootloader 最本职的两个工作:启动、加载。关于这个文件的解析,将在后面详细阐述。

#### 1.2 loader

关于为什么要加载(why)这个问题,这一小节就来细说。如果你对这一部分没有特别强烈的求知欲,可以暂时先跳过,以后再来阅读。我们的程序代码存放或者运行的地方称为存储介质。储存介质选择的主要参考:速度,尺寸,价格。存储介质按照不同的方法,可以分为不同的种类。

- 1)按存取方式分类如果存储器中任何存储单元的内容都能被随机存取,且存取时间和存储单元的物理位置无关,这种存储器称为随机存储器。半导体存储器和磁芯存储器都是随机存储器。如果存储器只能按某种顺序来存取,也就是说存取时间和存储单元的物理位置无关,这种存储器称为顺序存储器。例如,磁带存储器就是顺序存储器。一般来说,顺序存储器的存取周期较长。磁盘存储器是半顺序存储器。
- 2) 按存储器的读写功能分类有些半导体存储器存储的内容是固定不变的,即只能读出而不能写入,因此这种半导体存储器称为只读存储器 (ROM)。既能读出又能写入的半导体存储器,称为随机存储器 (RAM)。
- 3) 按信息的可保存性分类断电后信息即消失的存储器, 称为易失性存储器, 或者非永久记忆的存储器。断电后仍能保存信息的存储器, 称为非易失性存储器, 或永久性记忆的存储器。磁性材料做成的存储器是永久性存储器, 半导体读写存储器 RAM 是非永久性存储器。

我们常见的储存介质大类有:磁带,硬盘,ROM,RAM,具体到嵌入式:经常用到是 norflash,nandflash,sdcard,TF卡,compact卡,sdram,ram等。为什么会出现这么多种类?这个是价格和需求平衡的结果。比如,我们知道程序最后运行必须要有随机可读写存储器来存储变量,且速度要快,这个导致了RAM的产生,但是RAM价格昂贵,又导致了sdram的产生,sdram和ram的区别就是它是靠电容的

值来保存 0, 1 信息,时间一长就会丢失数据,故需要周期性刷新,这个在 sdram 控制器芯片的控制下能很好解决,且不太影响性能,但是它速度比 ram 低一些,且复杂些,但是价格低很多,且容易做到很大,故是一种很好的储存器,因此目前无论是嵌入式还是 pc 设备都广泛使用到了 sdram。虽然 sdram 解决了可读写问题,且速度问题。但是它们都是非永久记忆的存储器,断电后信息即消失的存储器,明显不能满足我们要求永久保存我们代码的需求,你总不至于,每次启动电脑都要下载一次程序吧,于是就产生 nandflash, 硬盘这些永久记忆的存储器(硬盘太大,很少用在嵌入式系统中),这些储存器是永久,且能做到很大容量,但是速度慢,不过还是可以承受的,因为我们有办法解决这个问题?如何解决,就是前面说的加载,就是说在启动阶段,Bootloader 启动后就从这些储存介质拷贝程序到 sdram,这样真正运行时,代码和数据是从 sdram 中读取的,也就没有速度问题了,这也是为啥要 Bootloader 的原因。

有了 nandflash, 硬盘这些永久记忆的存储器还不够,为啥?因为它们是按块访问的,而不是按地址访问,这种块模式访问往往需要有硬件控制器,而硬件控制器又需要由程序来控制,那这个控制器的程序从何而来?这就是鸡生蛋,蛋生鸡的问题,正因为这样又出来一种存储器——ROM,比如 norflash, 只读存储器,这种储存器也是永久记忆的存储器,但它和 nandflash 等不一样,它是按地址随机访问的,也就是说不需要驱动,和 sdram 的访问方式一样,可以很简单的访问数据,这就解决了这个问题,但是这种按地址访问的永久记忆的存储器相比有点贵,且不能做到很大。其实也没必要过多的使用这中存储器,为啥?因为它是只读的,没法修改,不会过多使用,所以只要能够容下 Bootloader 这些程序就可以了,其他的代码交给廉价的可写的 nandflash 吧,当然对于代码还是可以一直放在 rom 中的,这样可以减少 Sdam 的使用。

也许你会说为啥不出产一种按地址访问的可读写永久记忆的存储器,是可以啊,但是代价太高,没必要,只要合理搭配,就可以满足需求,当然不排除有一天,按地址访问的可读写永久记忆的存储器很便宜了,但是这个世界没有完美的东西,优点越多,缺点也越多。

下面就来说下嵌入式储存器搭配问题: 硬盘肯定是不到万不得已,是不选择的,因为这个家伙体积大,功耗大,也不安静,不过对于需要储存上 10G 的数据的应用,还不得不用它。Bootloader 程序的储存器肯定得要是按地址随机存取的永久性记忆的存储器,当然对于支持 nandflash 启动的 soc,也可以储存在 nandflash,比如 s3c2410,2440,同时又比如 omap3530 是支持 sdcard 启动的,这样的 SOC 芯片也是可以将 Bootloader 放在 sdcard 上的。也许到了这里,你会有一种强烈的好奇

性?刚才不是说 nandflash,sdcard 都是需要控制器才能访问数据的啊,控制器又需 要程序,上面的 s3c2410,omap3530 等芯片是如何做到从这些地方启动的。其实解 决方法和我们上面探讨的一样,就是必须有一个拷贝动作,这个拷贝动作可以有 三种方式,一种是硬件方式,另一种是软件方式1)硬件方式:就是硬件实现储 存设备控制器的控制,读取指定大小的数据,它没法做到控制器的驱动程序那样, 可以随机读取任意大小的数据,但是只要能够拷贝指定地址指定大小的数据,就 已经够了,硬件可以看成是简化版的驱动 2) 软件方式:这个就更简单了,芯片自 带一个 ROM, 往往是片内 ROM, 这个 ROM 里装有驱动程序, 这个驱动程序负责 将我们的 Bootloader 从 nandflash 或 sdcard 等储存器拷贝到 sdram 或 ram 后,然后 跳到我们的 Bootloader 运行,这样其实和我们将 Bootloader 储存在 rom 是一样的, 只不过板子自带了一个 Bootloader, 这个简单的 Bootloader 先于我们的 Bootloader 运行,主要实现小量数据(至少包括我们的 Bootloader 的自我拷贝代码)拷贝。 其实还有另外一种启动技巧,那就是内存映射:就是说当用户使用跳线选择方 式后,硬件自动开启了内存映射,将其他内存地址映射到 cpu 启动地址,比如 pb11mpcore,cpu 的启动地址是 0x0,如果配置为 Norflash 启动,则可将 norflash 的 地址 0x40000000 0x43FFFFFF 映射到 0x0 0x3ffffff, 这样就相当于从 norflash 启动, 这种方式是经常用的方式。由于内存映射到地址 0x0 了,导致地址 0x0 对应的的 内存没法使用,因此启动后需将这个映射取消这种和上面的方式不同,这种需要 有按地址随机存取存储器的支持,即将储存启动代码的存储器的地址映射到启动 地址。

说完 Bootloader 的储存介质,就得说说操作系统(通常叫 kernel)映像文件的储存介质。嵌入式操作系统这类操作系统一般比较小,选择余地有很多,可以放在 rom 中,也可以放在 nandflash 中,因为不论放在哪里,只要 Bootloader 能找到,拷贝到 sdram 就可以了,所以关键看 Bootloader 是否强大,对于很强大的 Bootloader,其实操作系统都可以放在主机上,然后 Bootloader 可以通过网络将操作系统下载到 sdram,然后启动操作系统。对于 Bootloader 和内核链在一起的操作系统,操作系统肯定就是跟 Bootloader 一起储存在一种储存介质中了啊。

## 第二章 aCoral 启动

2440 上电之后,CPU 将从 0 地址处开始取指执行指令。如果将 aCoral 程序存放在 Norflash 中,并通过 mini2440 开关 S2 选择 Norflash 启动,则 Norflash 从硬件层面上被映射为 0 地址开始的一段地址;如果将 aCoral 放在 Nandflash 中,并且通过 S2 开关选择 Nandflash 启动,则开发板在上电后自动将 Nandflash 前 4KB 内容复制到开发板上的一块 SRAM 中。这块 SRAM 我们称为 Stepping Stone(垫脚石),并且 Stepping Stone 的地址就是从 0 地址开始。所以,不论选择何种启动方式,2440 都将从 0 地址开始执行第一行代码。

图2-1显示了复位后 S3C2440A 的存储器映射情况,详细请参考《S3C2440 中文手册》第五章。



图 2-1 复位后 S3C2440A 的存储器映射

## 2.1 启动-第一阶段

之前说到, aCoral 的 bootloader 其实就是

hal/s3c2440/src/start.S

我们将其称为 aCoral 的启动文件。启动文件中,这行跳转指令就是整个 aCoral 的入口,将被烧录在 Nandflash 或 Norflash 的 0 地址。

1 \_\_ENTRY:
2 b ResetHandler

这句跳转程序将跳转到 ResetHandler 标号处,执行一些上电之后的硬件初始 化工作,包括关闭看门狗、配置时钟、堆栈初始化、复制 OS 到 SDRAM 等。我们 一点点来看这些代码。

PS: 请准备好《S3C2440 中文手册》

### 2.1.1 禁用看门狗

1 @ disable watch dog timer
2 mov r1, #0x53000000
3 mov r2, #0x0
4 str r2, [r1]

看门狗 WatchDog 的名字形象的描述了它的工作原理,看门狗每隔一段时间(比如: 3 个小时)它就会饥饿,每次饥饿时都叫,如果不想让它叫,只要我们保证在 3 个小时内喂狗一次就行。因此我们要及时的对看门狗控制器执行喂狗操作。看门狗定时器内部有一个递减计数器,当该计数器递减为 0 的时候,就会自动重启控制器,如果我们写有这样的程序,该程序在定时器计数器递减为 0 之前,将其递减计数器重新设置一下(喂狗),那么就不会产生重启操作。假如机器设备出现异常情况下如死机,CPU 执行出错,程序跑飞等情况,CPU 就会陷入非正常的执行流程,就不会去执行重置计数器的程序,当计数器递减为 0 时,会产生复位控制器信号,机器就会重新启动,恢复正常执行流程。这样的设计原理就解决了很多环境恶劣的情况下,对服务器进行重启的任务。上面的重置倒计数的操作通常叫做"喂狗"。为了避免看门狗带来的影响,简化系统,我们选择关闭看门狗。

上述代码向地址 0x53000000 (r1), 也就是看门狗定时器控制寄存器 (WTCON) 写入了 0x0000 (r2), 即 16 个 0。结合图2-2, 可以知道,这样配置的结果就是禁止了看门狗,系统也就不需要定时去喂狗了。

当然了,在比较正式的系统中,看门狗是必须要开启的,防止系统一直死机。 这里由于我们只是在开发 aCoral,所以暂时关闭。

| 寄存器   | 地址         | R/W | 描述          | 复位值    |
|-------|------------|-----|-------------|--------|
| WTCON | 0x53000000 | R/W | 看门狗定时器控制寄存器 | 0x8021 |

| WTCON   | 位      | 描述                                                               | 初始状态 |
|---------|--------|------------------------------------------------------------------|------|
| 预分频值    | [15:8] | 预分频值。该值范围从 0 到 255 ( 2 <sup>8</sup> -1 )                         | 0x80 |
| 保留      | [7:6]  | 保留。正常工作中这两位必须为 00                                                | 00   |
| 看门狗定时器  | [5]    | 看门狗定时器的使能或禁止位<br>0 = 禁止                                          | 1    |
| 时钟选择    | [4:3]  | 全局闹钟使能<br>00:16                                                  | 00   |
| 中断产生    | [2]    | 中断的使能或禁止位<br>0 = 禁止                                              | 0    |
| 保留      | [1]    | 保留。正常工作中此位必须为0                                                   | 0    |
| 复位使能/禁止 | [0]    | 看门狗定时器复位输出的使能或禁止位<br>1:看门狗超时时发出 S3C2440A 复位信号<br>0:禁止看门狗定时器的复位功能 | 1    |

图 2-2 看门狗定时器控制 (WTCON) 寄存器 (手册第 18 章)

## 2.1.2 关中断

```
@ disable all interrupts
1
2
           mov
                   r1, #INT CTL BASE
3
           mov
                   r2, #0xffffffff
4
                   r2, [r1, #oINTMSK]
           str
                   r2 = 0 \times 7 ff
5
           ldr
6
           str
                   r2, [r1, #oINTSUBMSK]
```

系统刚上电启动,这个时候,aCoral 的中断系统还没有初始化,此时发生中断我们无法处理,所以要关闭中断。

代码中,立即数 INT\_CTL\_BASE = 0x4A000000,立即数 oINTMSK = 0x08,两个立即数相加的地址即指向中断屏蔽寄存器。中断屏蔽寄存器由 32 位组成,其每一位都都涉及一个中断源。如果某个指定为被设置为 1,则 CPU 不会去服务来自相应中断源(请注意即使在这种情况中,SRCPND 寄存器的相应位也设置为 1)的中断请求。如果屏蔽位为 0,则可以服务中断请求,如图2-3。

oINTSUBMSK 指向的次级中断屏蔽寄存器类似。

| 寄存器    | 地址             | R/W | 描述                     | 复位值       |
|--------|----------------|-----|------------------------|-----------|
| INTMSK | 0X4A000008     | R/W | 决定屏蔽哪个中断源。被屏蔽的中断源将不会服务 | 0xFFFFFFF |
|        | G/( I/ 1000000 |     | 0 = 中断服务可用             | 0         |

| INTMSK       | 位    |         | 描述     | 初始状态 |
|--------------|------|---------|--------|------|
| INT_ADC      | [31] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_RTC      | [30] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_SPI1     | [29] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_UART0    | [28] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_IIC      | [27] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_USBH     | [26] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_USBD     | [25] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_NFCON    | [24] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_UART1    | [23] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_SPI0     | [22] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_SDI      | [21] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_DMA3     | [20] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_DMA2     | [19] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_DMA1     | [18] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_DMA0     | [17] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_LCD      | [16] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_UART2    | [15] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_TIMER4   | [14] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_TIMER3   | [13] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_TIMER2   | [12] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_TIMER1   | [11] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_TIMER0   | [10] | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_WDT_AC97 | [9]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_TICK     | [8]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| nBATT_FLT    | [7]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| INT_CAM      | [6]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| EINT8_23     | [5]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| EINT4_7      | [4]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| EINT3        | [3]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| EINT2        | [2]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| EINT1        | [1]  | 0 = 可服务 | 1 = 屏蔽 | 1    |
| EINT0        | [0]  | 0 = 可服务 | 1 = 屏蔽 | 1    |

图 2-3 中断屏蔽寄存器 (手册第 14 章)

## 2.2 时钟初始化

```
1     @ initialise system clocks
2     mov     r1 , #CLK_CTL_BASE
3     mvn     r2 , #0xff000000
4     str     r2 , [r1 , #oLOCKTIME]
5
6     mov     r1 , #oCLKDIVN
```

| 7  | mov | r2 , #M_DIVN             |                       |
|----|-----|--------------------------|-----------------------|
| 8  | str | r2 , [r1 , #oCLKDIVN]    |                       |
| 9  |     |                          |                       |
| 10 | mrc | p15, 0, r1, c1, c0, 0    | @ read ctrl register  |
| 11 | orr | r1, $r1$ , $#0xc0000000$ | @ Asynchronous        |
| 12 | mcr | p15, 0, r1, c1, c0, 0    | @ write ctrl register |
| 13 |     |                          |                       |
| 14 | mov | r1, #CLK_CTL_BASE        |                       |
| 15 | ldr | r2, =vMPLLCON            | @ clock user set      |
| 16 | str | r2, [r1, #oMPLLCON]      |                       |

mini2440 上电后,系统工作在板载晶振 12MHz 的频率下。这个频率比较低,系统的性能还没有完全得到发挥,所以需要激发一下潜能,使用一种叫做锁相环PLL 的东西来倍频,加倍之后的频率再给 CPU 和板子上的其他设备使用。

mini2440 有两个锁相环,MPLL 和 UPLL。MPLL 的输出频率直接给 CPU 使用,称为 FCLK,同时经过上面初始化系统时钟后得到另外两个频率 HCLK 和 PCLK,分别给板载高速硬件和低速外设使用。

将 M\_DIVN = 0x5 写入 oCLKDIVN + oCLKDIVN = 0x4C000014 寄存器后,三 者的比例为 FCLK: HCLK: PCLK=8: 2: 1,如图2-4

| 寄存器     | 地址         | R/W | 描述        | 复位值        |
|---------|------------|-----|-----------|------------|
| CLKDIVN | 0x4C000014 | R/W | 时钟分频控制寄存器 | 0x00000004 |

| CLKDIVN    | 位        | 描述                                   | 初始状态 |
|------------|----------|--------------------------------------|------|
|            |          | UCLK 选择寄存器 ( UCLK 必须为 48MHz 给 USB )  |      |
| DIVN UPLL  | [3]      | 0: UCLK = UPLL 时钟                    | 0    |
| DIVIN_OPEL | [9]      | 当 UPLL 时钟被设置为 48MHz 时,设置为 0          |      |
|            |          | 当 UPLL 时钟被设置为 96MHz 时,设置为 1          |      |
|            |          | 00 : HCLK = FCLK/1                   |      |
|            | VN [2:1] | 01 : HCLK = FCLK/2                   |      |
| HDIVN      |          | 10: HCLK = FCLK/4 当 CAMDIVN[9] = 0 时 | 00   |
| HOIVIN     |          | HCLK = FCLK/8 当 CAMDIVN[9] = 1 时     | 00   |
|            |          | 11: HCLK = FCLK/3 当 CAMDIVN[8] = 0 时 |      |
|            |          | HCLK = FCLK/6 当 CAMDIVN[8] = 1 时     |      |
| PDIVN      | [0]      | 0: PCLK 是和 HCLK/1 相同的时钟              | 0    |
|            | [0]      | 1:PCLK 是和 HCLK/2 相同的时钟               | 0    |

图 2-4 时钟分频控制寄存器 (手册第7章)

将 vMPLLCON = 0x1FC0021 写入 oMPLLCON + CLK\_CTL\_BASE = 0x4C000004 寄存器后,得到 FCLK≈400MHz,再根据之前的比例得到 HCLK 和 PCLK 分别为 100MHz 和 50MHz。计算过程见图2-5,其中 Fin 就是晶振的频率 =12MHz。 支方路 With P/W

| 寄存器     | 地址         | R/W | 描述         | 复位值        |  |
|---------|------------|-----|------------|------------|--|
| MPLLCON | 0x4C000004 | R/W | MPLL 配置寄存器 | 0x00096030 |  |
| UPLLCON | 0x4C000008 | R/W | UPLL 配置寄存器 | 0x0004d030 |  |

| PLLCON | 位       | 描述     | 初始状态        |
|--------|---------|--------|-------------|
| MDIV   | [19:12] | 主分频器控制 | 0x96 / 0x4d |
| PDIV   | [9:4]   | 预分频器控制 | 0x03 / 0x03 |
| SDIV   | [1:0]   | 后分频器控制 | 0x0 / 0x0   |

#### 注意:

当你设置 MPLL 和 UPLL 的值时,你必须首先设置 UPLL 值再设置 MPLL 值。(大约需要 7 个 NOP 的间隔)

#### MPLL 控制寄存器

```
MpII = (2 \times m \times Fin) / (p \times 2^{s})
m = (MDIV + 8), p = (PDIV + 2), s = SDIV
```

图 2-5 PLL 控制寄存器 (手册第7章)

UPLL 的输出频率没有配置,直接给 USB 使用。

## 2.2.1 存储空间初始化

| 1  | bl        | memsetup                 |
|----|-----------|--------------------------|
| 2  |           |                          |
| 3  | @****     | ******                   |
| 4  | @ init    | ialise the static memory |
| 5  | @ set     | memory control registers |
| 6  | @****     | *********                |
| 7  | memsetup: |                          |
| 8  | mov       | r1 , #MEM_CTL_BASE       |
| 9  | adrl      | r2, mem_cfg_val          |
| 10 | add       | r3 , r1 , #52            |
| 11 | 1: 1dr    | r4 , [r2], #4            |
| 12 | str       | r4 , [r1], #4            |
| 13 | cmp       | r1 , r3                  |
| 14 | bne       | 1 b                      |
| 15 | mov       | pc, 1r                   |

memsetup 就是初始化一下 mini2440 的存储器 (BANK0 BANK7), 具体自行 阅读《S3C2440 中文手册》第 5 章。

## 2.2.2 栈初始化

| 1 | bl | InitStacks |
|---|----|------------|
| 2 |    |            |

```
3
           @************
4
                           堆栈初始化
5
           @************
6
   InitStacks:
7
8
            mov r2, lr
9
                    r0, cpsr
            mrs
                    r0, r0, #MODE MASK
10
            bic
11
            orr
                    r1, r0, #UND MODE | NOINT
                                              @UndefMode
12
                     cpsr cxsf, r1
            msr
                     sp,=UDF stack
                                              @ UndefStack=0x33FF 5C00
13
            ldr
14
15
                     r1, r0, #ABT MODE | NOINT
            orr
                                              @AbortMode
16
                     cpsr cxsf, rl
            msr
17
                     sp,=ABT stack
                                              @ AbortStack=0x33FF 6000
            ldr
18
19
                     r1, r0, #IRQ MODE | NOINT
            orr
20
                     cpsr cxsf, r1
                                              @IRQMode
            msr
21
                     sp,=IRQ stack
                                              @ IRQStack=0x33FF 7000
            1 d r
22
23
                     r1, r0, #FIQ MODE | NOINT
            orr
24
            msr
                     cpsr_cxsf, r1
                                              @FIQMode
25
            ldr
                     sp,=FIQ stack
                                              @ FIQStack=0x33FF 8000
26
27
            bic
                     r0, r0, #MODE MASK | NOINT
                     r1, r0, #SVC MODE
28
            orr
29
            msr
                     cpsr cxsf, r1
                                              @SVCMode
30
                     sp,=SVC stack
                                              @ SVCStack=0x33FF 5800
            ldr
31
32
            mrs
                     r0, cpsr
33
            bic
                    r0, r0, #MODE MASK
34
                    r1, r0, #SYS_MODE | NOINT
            orr
                     cpsr cxsf, r1
35
                                              @ userMode
            msr
36
                     sp,=SYS stack
            ldr
37
38
            mov
                     pc, r2
```

InitStacks 就是初始化一下 arm 处理器各个模式的栈。因为影子寄存器的存在,

每个模式的 sp 寄存器都是独立的(除了系统和用户模式)。具体的做法就是通过 修改 cpsr 寄存器的 [5:0] 位,进入每一个模式并修改 sp 寄存器位该模式的栈底地址。每个模式的栈底地址定义于链接脚本。链接脚本将在下一节进行讲解。cpsr 寄存器各位编排如图2-6。



图 2-6 程序状态寄存器格式 (手册第 2 章)

PS: 关于影子寄存器(也称分组寄存器),请查看《S3C2440中文手册》第2章图 2-3。

## 2.2.3 自我拷贝(loader)

```
adr r0, ENTRY
1
2
            ldr rl, text start
            cmp r0, r1
3
            blne copy self
4
5
6
   copy self:
7
8
            1dr
                    r1, = (4 << 28) |(2 << 4)|(3 << 2) /* address of
               Internal SRAM 0x4000002C*/
                    r0, #0
9
            mov
                    r0, [r1]
10
            str
11
12
13
                    r1, #0x2c
                                    /* address of men 0x0000002C*/
            mov
                    r0, [r1]
14
            ldr
                    r0, #0
15
            cmp
16
            bne
                    copy from rom
17
```

这一步非常重要。如果 aCoral 是烧写在 nor flash 中,启动时是从 0 地址开始运行的,那 aCoral 怎么到内存 SDRAM(起始地址为 0x30000000)中运行呢?这项任务是由 aCoral 自己来完成的,也就是我搬起了我自己。可以看到 copy\_self 的执行条件

#### 2.2.4 bss 段清零

```
1
          ldr
               r0, bss start
2
          1 d r
               r1, _bss_end
3
                mem clear
          bl
4
          @*************
5
6
          @ clear memory
7
          @ r0: start address
          @ rl: length
8
9
          @**********
10
11
   mem clear:
12
          mov r2,#0
13
   1:
          str r2,[r0],#4
          cmp r0, r1
14
15
          blt 1b
16
          mov pc, 1r
```

bss 段中的数据都是未初始化或者初始值为 0 的,所以清空 bss 段就是将其中的内存全部赋值为 0。mem\_clear 的两个参数分别为 bss 段的起始地址和结束地址,都定义在链接脚本中。

## 2.2.5 跳转至下一阶段

```
1 ldr pc,=acoral_start
```

代码最终将寄存器 pc 设置为 acoral\_start 的值,表示 CPU 将跳转到 acoral\_start 函数处执行。acoral\_start 函数位于

kernel\include\core.c

## 2.3 启动-第二阶段

core.c 中的 acoral\_start() 函数如下

这部分代码中,最重要的就是 acoral\_module\_init() 这个函数。这个函数将初始化 aCoral 的各个模块,包括中断、内存、线程等嵌入式操作系统必需的模块。初始化完成后,执行 acoral\_core\_cpu\_start() 函数,aCoral 开始运行。

## 第三章 链接脚本